import random
import re
from datetime import datetime

from flask import request, abort, current_app, make_response, jsonify

from info import redis_store, constants, db
from info.libs.yuntongxun.sms import CCP
from info.models import User
from info.utils.captcha.captcha import captcha
from info.utils.response_code import RET
from flask import session

from . import passport_blu


@passport_blu.route("/register", methods=["POST"])
def register():
    """
    注册的逻辑
    1. 获取参数
    2. 校验参数
    3. 取到服务器保存的真实的短信验证码内容
    4. 校验用户输入的短信验证码内容和真实验证码内容是否一致
    5. 如果一致, 初始化 User 模型, 并且赋值属性
    6. 将 User 模型添加到数据库
    7.返回响应
    :return:
    """
    # 1.获取参数
    param_dict = request.json
    mobile = param_dict.get("mobile")
    smscode = param_dict.get("smscode")
    password = param_dict.get("password")

    # 2. 校验参数
    if not all([mobile, smscode, password]):
        return jsonify(errno=RET.PARAMERR, errmsg="参数不够")
    # 校验手机号码是否正确
    if not re.match('^1[35678]\\d{9}', mobile):
        return jsonify(errno=RET.PARAMERR, errmsg="手机号格式不正确")

    # 3. 取到服务器保存的真实的短信验证码内容
    try:
        real_sms_code = redis_store.get("SMS_" + mobile)
    except Exception as f:
        current_app.logger.error(f)
        return jsonify(errno=RET.DBERR, errmsg="数据查询失败")
    if not real_sms_code:
        return jsonify(errno=RET.NODATA, errmsg="验证码已过期")

    # 4. 校验用户输入的短信验证码内容和真实验证码内容是否一致
    if real_sms_code != smscode:
        return jsonify(errno=RET.DATAERR, errmsg="验证码输入错误")

    # 5. 如果一致, 初始化 User 模型, 并且赋值属性
    user = User()
    user.mobile = mobile
    # 暂时没有昵称, 使用手机号代替
    user.nick_name = mobile
    # 记录用户最后登陆时间
    user.last_login = datetime.now()
    # 对密码做处理
    # 需求 : 在设置 password 的时候, 去对 password 进行加密, 并且将加密结果 给 User, password_hash 赋值
    user.password = password

    # 6. 将 User 模型添加到数据库
    try:
        db.session.add(user)
        db.session.commit()
    except Exception as f:
        current_app.logger.error(f)
        db.session.rollback()
        return jsonify(errno=RET.DBERR, errsmg="数据保存失败")
    # 往 session 中保存数据表示当前已经登录成功
    session["user_id"] = user.id
    session["mobile"] = user.mobile
    session["nick_name"] = user.nick_name

    # 7.返回响应
    return jsonify(errno=RET.OK, errmsg="注册成功")


@passport_blu.route('/login', methods=['POST'])
def login():
    """
    登录主页逻辑
    1. 获取参数
    2. 校验参数
    3. 校验密码是否正确
    4. 保存用户登录状态
    5. 响应
    :return:
    """
    # 1. 获取参数
    json_data = request.json
    mobile = json_data.get("mobile")
    password = json_data.get("password")

    # 2. 校验参数
    if not all([mobile, password]):
        return jsonify(errno=RET.PARAMERR, errsmg="参数不足")
    # 2.1 校验手机号
    if not re.match('^1[235678]\\d{9}|admin', mobile):
        return jsonify(errno=RET.PARAMERR, errmsg="手机号格式不正确")

    # 2. 从数据库查询出指定的用户
    try:
        user = User.query.filter(User.mobile == mobile).first()
    except Exception as f:
        current_app.logger.debug(f)
        return jsonify(errno=RET.DBERR, errmsg="数据查询错误")

    # 判断用户是否存在
    if not user:
        return jsonify(errno=RET.NODATA, errmsg="用户不存在")

    # 3. 校验密码
    if not user.check_password(password):
        return jsonify(erron=RET.PWDERR, errmsg="密码错误")

    # 4. 保存用户登录状态
    session["user_id"] = user.id
    session["mobile"] = user.mobile
    session["nick_name"] = user.nick_name

    # 设置当前用户最后一次登录时的间
    user.last_login = datetime.now()

    # 如果在视图函数中,对模型身上的模型属性有修改,那么需要 commit 到数据库保存
    # 但是可以不用自己去写 db.session.commit(), 前提是对 SQLAlchemy 有过相关的配置

    # try:
    #     db.session.commit()
    # except Exception as f:
    #     db.session.rollback()
    #     current_app.logger.error(f)

    # 5. 响应
    return jsonify(errno=RET.OK, errmsgp="登陆成功")


@passport_blu.route('/logout')
def logout():
    """
    退出登录
    1. 删除 redis 的用户信息
    2. 刷新页面
    :return:
    """
    # pop 是移除session中的数据(dict)
    # pop 会有一个返回值, 如果移除的key不存在,就会返回None
    session.pop("user_id", None)
    session.pop("mobile", None)
    session.pop("nick_name", None)
    # 如果不清除 is_admin 先登陆管理员,后登陆普通用户,又能访问到管理员页面
    session.pop('is_admin', None)
    return jsonify(errno=RET.OK, errmsg="退出成功")


@passport_blu.route('/sms_code', methods=['POST'])
def send_sms_code():
    """
    发送短信的逻辑
    1. 获取参数: 手机号, 图片验证码内容, 图片验证码编号(随机值)
    2. 校验参数(参数是否符合规则, 判断是否有值)
    3. 先从redis中取出真实的验证内容
    4. 与用户的验证码内容进行对比, 如果对比不一致, 那么返回验证码输入错误
    5. 如果一致, 生成验证码的内容(随机数据)
    6. 发送短信验证码
    7. 告知发送结果
    :return:
    """
    '{"mobiel":"1881111111", "image_code":"AAAA", "image_code_id":"u233dscsjksjvvvnnvnvvn"}'
    # 1. 获取参数: 手机号, 图片验证码内容, 图片验证的编号(随机值)
    # params_dict = json.load(request.data)
    params_dict = request.json

    mobile = params_dict.get("mobile")
    image_code = params_dict.get("image_code")
    image_code_id = params_dict.get("image_code_id")

    # 2. 校验参数(参数是否符合规则, 判断是否有值)
    # 判断是否有值
    if not all([mobile, image_code, image_code_id]):
        # {"error":4100, "errmsg": "参数有误"}
        return jsonify(errno=RET.PARAMERR, errmsg="参数有误")
    if not re.match('^1[235678]\\d{9}', mobile):
        return jsonify(errno=RET.PARAMERR, errmsg="手机号格式不正确")
    # 3. 先从redis中取出真实的验证内容
    try:
        real_image_code = redis_store.get("ImageCodeId" + image_code_id)
        print(real_image_code)
    except Exception as f:
        current_app.lopper.error(f)
        return jsonify(errno=RET.DBERR, errmsg="数据查询失败")

    if not real_image_code:
        return jsonify(errno=RET.NODATA, errmsg="图片验证码已过期")
    # 4.与用户的验证码内容进行对比, 如果对比不一致, 那么返回验证码输入错误
    if real_image_code.upper() != image_code.upper():
        return jsonify(errno=RET.DATAERR, errmsg="验证码输入错误")
    # 5. 如果一致, 生成验证码的内容(随机数据)
    # 随机数字, 保证数据长度为6位,不够在前面补上0
    sms_code_str = "%06d" % random.randint(0, 999999)
    current_app.logger.debug("短信验证码内容是: %s " % sms_code_str)
    # 6. 发送短信验证码
    # result = CCP().send_template_sms(mobile, [sms_code_str, constants.SMS_CODE_REDIS_EXPIRES / 5], "1")
    # print(result)
    # if result != 0:
    #     # 代表发送不成功
    #     return jsonify(errno=RET.THIRDERR, errmsg="发送失败")

    # 保存验证码内容到redis中
    try:
        redis_store.set("SMS_" + mobile, sms_code_str, constants.SMS_CODE_REDIS_EXPIRES)
    except Exception as f:
        current_app.logger.error(f)
        return jsonify(errno=RET.DBERR, errmsg="数据保存失败")

    # 7. 告知发送结果
    return jsonify(errno=RET.OK, errmsgp="发送成功")


@passport_blu.route('/image_code')
def get_image_code():
    """
    生成验证码并返回

    1. 取到参数
    2. 判断参数是否有值
    3. 生成图片验证码
    4. 保存图片验证码文字内容到redis
    5. 返回验证码图片
    6. 设置数据类型, 以便浏览器更加方便识别数据类型
    : return:
    """
    # 1. 取到参数
    # args: 取到url中 ? 后面的参数
    image_code_id = request.args.get("imageCodeId", None)

    # 2. 判断参数是否有值
    if not image_code_id:
        return abort(403)
    # 3. 生成图片验证码
    name, text, image = captcha.generate_captcha()
    # 4. 保存图片验证码文字内容到redis
    try:
        redis_store.set("ImageCodeId" + image_code_id, text, constants.IMAGE_CODE_REDIS_EXPIRES)
    except Exception as f:
        current_app.logger.error(f)
        abort(500)
    # 5. 返回验证码图片
    response = make_response(image)
    # 6. 设置数据类型, 以便浏览器更加方便识别数据类型
    response.headers['Content-Type'] = 'image/jpg'
    return response
